// *****************************************************************************
// Copyright (C) 2025 EclipseSource GmbH.
//
// This program and the accompanying materials are made available under the
// terms of the Eclipse Public License v. 2.0 which is available at
// http://www.eclipse.org/legal/epl-2.0.
//
// This Source Code may also be made available under the following Secondary
// Licenses when the conditions for such availability set forth in the Eclipse
// Public License v. 2.0 are satisfied: GNU General Public License, version 2
// with the GNU Classpath Exception which is available at
// https://www.gnu.org/software/classpath/license.html.
//
// SPDX-License-Identifier: EPL-2.0 OR GPL-2.0-only WITH Classpath-exception-2.0
// *****************************************************************************

import { MutableChatRequestModel } from '@theia/ai-chat';
import { ToolProvider, ToolRequest } from '@theia/ai-core';
import { inject, injectable, optional } from '@theia/core/shared/inversify';
import { LIST_CHAT_CONTEXT_FUNCTION_ID, RESOLVE_CHAT_CONTEXT_FUNCTION_ID, UPDATE_CONTEXT_FILES_FUNCTION_ID } from '../common/context-functions';
import { FILE_VARIABLE } from '@theia/ai-core/lib/browser/file-variable-contribution';
import { ContextFileValidationService, FileValidationState } from '@theia/ai-chat/lib/browser/context-file-validation-service';

@injectable()
export class ListChatContext implements ToolProvider {
    static ID = LIST_CHAT_CONTEXT_FUNCTION_ID;

    getTool(): ToolRequest {
        return {
            id: ListChatContext.ID,
            name: ListChatContext.ID,
            description: 'Returns the list of context elements (such as files) specified by the user manually as part of the chat request.',
            handler: async (_: string, ctx: MutableChatRequestModel): Promise<string> => {
                if (ctx?.response?.cancellationToken?.isCancellationRequested) {
                    return JSON.stringify({ error: 'Operation cancelled by user' });
                }
                const result = ctx.context.variables.map(contextElement => ({
                    id: contextElement.variable.id + contextElement.arg,
                    type: contextElement.variable.name
                }));
                return JSON.stringify(result, undefined, 2);
            },
            parameters: {
                type: 'object',
                properties: {}
            },
        };
    }
}

@injectable()
export class ResolveChatContext implements ToolProvider {
    static ID = RESOLVE_CHAT_CONTEXT_FUNCTION_ID;

    getTool(): ToolRequest {
        return {
            id: ResolveChatContext.ID,
            name: ResolveChatContext.ID,
            description: 'Returns the content of a specific context element (such as files) specified by the user manually as part of the chat request.',
            parameters: {
                type: 'object',
                properties: {
                    contextElementId: {
                        type: 'string',
                        description: 'The id of the context element to resolve.'
                    }
                },
                required: ['contextElementId']
            },
            handler: async (args: string, ctx: MutableChatRequestModel): Promise<string> => {
                if (ctx?.response?.cancellationToken?.isCancellationRequested) {
                    return JSON.stringify({ error: 'Operation cancelled by user' });
                }

                const { contextElementId } = JSON.parse(args) as { contextElementId: string };
                const variable = ctx.context.variables.find(contextElement => contextElement.variable.id + contextElement.arg === contextElementId);
                if (variable) {
                    const result = {
                        type: variable.variable.name,
                        ref: variable.value,
                        content: variable.contextValue
                    };
                    return JSON.stringify(result, undefined, 2);
                }
                return JSON.stringify({ error: 'Context element not found' }, undefined, 2);
            }
        };
    }
}

@injectable()
export class AddFileToChatContext implements ToolProvider {
    static ID = UPDATE_CONTEXT_FILES_FUNCTION_ID;

    @inject(ContextFileValidationService) @optional()
    protected readonly validationService: ContextFileValidationService | undefined;

    getTool(): ToolRequest {
        return {
            id: AddFileToChatContext.ID,
            name: AddFileToChatContext.ID,
            parameters: {
                type: 'object',
                properties: {
                    filesToAdd: {
                        type: 'array',
                        description: 'The absolute paths of files to add to the context of the current chat.',
                        items: { type: 'string' }
                    }
                },
                required: ['filesToAdd']
            },
            description: 'Adds one or more files to the context of the current chat session. ' +
                'Only files that exist within the workspace boundaries will be added. ' +
                'Files outside the workspace or non-existent files will be rejected. ' +
                'Returns a detailed status for each file, including which were successfully added and which were rejected with reasons.',
            handler: async (arg: string, ctx: MutableChatRequestModel): Promise<string> => {
                if (ctx?.response?.cancellationToken?.isCancellationRequested) {
                    return JSON.stringify({ error: 'Operation cancelled by user' });
                }

                const { filesToAdd } = JSON.parse(arg) as { filesToAdd: string[] };

                const added: string[] = [];
                const rejected: Array<{ file: string; reason: string; state: string }> = [];

                for (const file of filesToAdd) {
                    if (this.validationService) {
                        const validationResult = await this.validationService.validateFile(file);

                        if (validationResult.state === FileValidationState.VALID) {
                            ctx.session.context.addVariables({ arg: file, variable: FILE_VARIABLE });
                            added.push(file);
                        } else {
                            rejected.push({
                                file,
                                reason: validationResult.message || 'File validation failed',
                                state: validationResult.state
                            });
                        }
                    } else {
                        ctx.session.context.addVariables({ arg: file, variable: FILE_VARIABLE });
                        added.push(file);
                    }
                }

                return JSON.stringify({
                    added,
                    rejected,
                    summary: {
                        totalRequested: filesToAdd.length,
                        added: added.length,
                        rejected: rejected.length
                    }
                });
            }
        };
    }
}
